home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Libris Britannia 4
/
science library(b).zip
/
science library(b)
/
DTP
/
DTP_TEX
/
3239.ZIP
/
DVIMTP.ZIP
/
DVIMTP.C
< prev
next >
Wrap
C/C++ Source or Header
|
1990-01-31
|
39KB
|
1,134 lines
/* -*-C-*- dvimtp.c */
/*-->dvimtp*/
/**********************************************************************/
/************************** dvijep ** or **dvimtp**********************/
/**********************************************************************/
#include "dvihead.h"
/***********************************************************************
***********************************************************************/
/**********************************************************************/
/************************ Device Definitions ************************/
/**********************************************************************/
/* All output-device-specific definitions go here. This section must
be changed when modifying a dvi driver for use on a new device */
#undef HPJETPLUS
#define HPJETPLUS 1 /* conditional compilation flag */
/* Comment out for Microsoft C command line definition
#undef MT905
#define MT905 1
*/
#define VERSION_NO "2.10" /* DVI driver version number */
#ifdef MT905
#define DEVICE_ID "Mannesmann Tally MT905 Laser Printer"
/* this string is printed at runtime */
#define OUTFILE_EXT "mtp"
#else
#define DEVICE_ID "Hewlett-Packard LaserJet Plus laser printer"
/* this string is printed at runtime */
#define OUTFILE_EXT "jep"
#endif
#define BYTE_SIZE 8 /* output file byte size */
#undef STDRES
#define STDRES 1 /* to get standard font resolution */
#define MAXLINE 4096 /* maximum input file line size */
/* it is needed only in special() */
#define MAXPAGEFONTS 16 /* silly limit on fonts/page */
#define XDPI 300 /* horizontal dots/inch */
#define XPSIZE 8 /* horizontal paper size in inches */
#define XSIZE (((XDPI*XPSIZE+2*HOST_WORD_SIZE-1)/\
(2*HOST_WORD_SIZE))*(2*HOST_WORD_SIZE))
/* number of horizontal dots; */
/* MUST BE multiple of */
/* 2*HOST_WORD_SIZE */
#define YDPI 300 /* vertical dots/inch */
#define YPSIZE 11 /* vertical paper size in inches */
#define YSIZE (YDPI*YPSIZE) /* number of vertical dots */
#define XORIGIN 30 /* measured pixel coordinates of */
#define YORIGIN -15 /* page origin (should be at (0,0) */
/***********************************************************************
Define some useful shorthand macros for all required LaserJet Plus
commands. With the exception of the CREATEFONT and DOWNLOADCHAR macros,
which require extra data, NO escape sequences appear anywhere else in
the text.
The LaserJet Plus allows coordinates in dots as well as the decipoints
supported by the regular LaserJet. Dot coordinates are more convenient,
so we use them exclusively. The Plus puts the page origin in the upper-
left corner, while we have it in the lower-left corner, so MOVEY adjusts
the coordinate accordingly.
With 7-bit fonts, only characters 33..127 are printable, and with 8-bit
fonts, 33..127 and 160..255 are printable. Since TeX assumes characters
0..127 are available, we remap character numbers in 0..32 to 160..192
and use 8-bit fonts. The macros conceal the remapping.
Whenever possible, escape sequences are combined and 0 parameters
omitted, in order to compress the output.
***********************************************************************/
/* Begin raster graphics .. {transfer raster graphics} .. end raster graphics */
#define BEGIN_RASTER_GRAPHICS OUTS("\033*t300R\033*r1A")
/* CREATE_FONT must be followed by a 26-byte font header (see readfont) */
#define CREATE_FONT OUTS("\033\051s26W")
#define DELETE_ALL_FONTS OUTS("\033*cF")
/* DOWNLOAD_CHARACTER is not a complete specification---it must be followed by
two-byte values of xoff, yoff, width, height, deltax */
#define DOWNLOAD_CHARACTER(nbytes) OUTF("\033\050s%dW",(nbytes)+16)
/* End raster graphics */
#define END_RASTER_GRAPHICS OUTS("\033*rB")
/* Map characters (0..32,33..127) to (160..192,33..127) */
#define MAP_CHAR(c) (((c) > 32) ? (c) : ((c) + 160))
/* Move absolute to (x,y), page origin in lower left corner */
#define MOVETO(x,y) OUTF2("\033*p%dx%dY",(x),YSIZE-(y))
/* Move absolute to (x,current y), page origin in lower left corner */
#define MOVEX(x) OUTF("\033*p%dX",(x))
/* Move absolute to (current x,y), page origin in lower left corner */
#define MOVEY(y) OUTF("\033*p%dY",YSIZE-(y))
/* Output TeX character number with mapping to HP LaserJet Plus number */
#define OUT_CHAR(c) OUTC(MAP_CHAR(c))
/* Output a 16-bit binary number in two bytes */
#define OUT16(n) {OUTC((n)>>8); OUTC((n));}
/* Eject a page from the printer--this is the last command on each page */
#define PAGEEJECT OUTC('\f')
/* Reset printer at job start. <ESC>E is global reset which clears
everything but permanent fonts, which we do not use. <ESC>&l0E resets
left margin to 0. <ESC>&a0L resets top margin to 0. Without these,
default margins are supplied which lie inside page. Unfortunately,
these do not permit negative values, so we have to further correct the
origin with (XORIGIN, YORIGIN) defined above; these bias (lmargin,
tmargin) set in dviinit(). */
#define RESET_PRINTER OUTS("\033E\033&lE\033&aL")
/* Move relative by (delx,dely) from current point */
#define RMOVETO(delx,dely) {\
if ((delx) == 0)\
RMOVEY(dely)\
else if ((dely) == 0)\
RMOVEX(delx)\
else\
OUTF2("\033*p%c%dx%c%dY",\
((delx) > 0 ? '+' : '-'),ABS(delx),\
((dely) < 0 ? '+' : '-'),ABS(dely));\
}
/* Move relative to (delx+currentx,currenty), page origin in lower left
corner */
#define RMOVEX(delx) {\
if ((delx) > 0)\
OUTF("\033*p+%dX",(delx));\
else if ((delx) < 0)\
OUTF("\033*p%dX",(delx));\
}
/* Move relative to (currentx,dely+currenty), page origin in lower left
corner */
#define RMOVEY(dely) {\
if ((dely) > 0)\
OUTF("\033*p%dY",-(dely));\
else if ((dely) < 0)\
OUTF("\033*p+%dY",-(dely));\
}
/* Round a POSITIVE floating-point value to the nearest integer */
#define ROUND(x) ((int)(x + 0.5))
/* Output a new rule at TeX position (x,y). The device coordinates will
be changed on completion. The rule origin is the TeX convention of the
lower-left corner, while the LaserJet Plus uses the upper-left corner.
*/
#define RULE(x,y,width,height) {\
MOVETO(x,(y)+height-1);\
OUTF2("\033*c%da%dbP",width,height);\
}
/* Set rule of same size as previous one at TeX position (x,y). The
device coordinates will be changed on completion. */
#define RULE2(x,y) {\
MOVETO(x,(y)+rule_height-1);\
OUTS("\033*cP");\
}
/* Set the current font and character in preparation for a DOWNLOAD_CHARACTER */
#define SET_CHAR_CODE(fontnumber,ch) {\
if (fontnumber)\
OUTF2("\033*c%dd%dE",fontnumber,MAP_CHAR(ch));\
else\
OUTF("\033*cd%dE",MAP_CHAR(ch));\
}
/* Set the number of copies of the current page (issue just before
PAGEEJECT) */
#define SETCOPIES(n) OUTF("\033&l%dX",MAX(1,MIN(n,99)))
/* Set the font number in preparation for a CREATE_FONT */
#define SET_FONT_ID(fontnumber) OUTF("\033*c%dd4F",(fontnumber));
#define TRANSFER_RASTER_GRAPHICS(nbytes) OUTF("\033*b%dW",(nbytes));
#define UNDEFINED_FONT 65535 /* fonts are numbered 0,1,2,... */
/* In order to handle the MAXPAGEFONTS restriction, we keep a set of the
fonts in use, accessing them with these two functions. If a later
printer model increases MAXFONTS beyond 32, we can transparently change
to another representation, but for now, we use an efficient bit
set implementation. */
UNSIGN32 page_members; /* the bit set for current page */
#define ADD_MEMBER(set,member) {set |= (1 << (member));}
#define IS_MEMBER(set,member) (set & (1 << (member)))
#define EMPTY_SET(set) {set = 0L;}
#include "main.h"
#include "abortrun.h"
#include "actfact.h"
#include "alldone.h"
/*-->bopact*/
/**********************************************************************/
/******************************* bopact *******************************/
/**********************************************************************/
void
bopact() /* beginning of page action */
{
page_fonts = 0; /* no fonts used on this page yet */
EMPTY_SET(page_members);
rule_height = -1; /* reset last rule parameters */
rule_width = -1;
str_ycp = -1; /* reset string y coordinate */
}
#include "chargf.h"
#include "charpk.h"
#include "charpxl.h"
#include "clrrow.h"
#include "dbgopen.h"
/*-->devinit*/
/**********************************************************************/
/****************************** devinit *******************************/
/**********************************************************************/
void
devinit(argc,argv) /* initialize device */
int argc;
char *argv[];
{
RESET_PRINTER; /* printer reset */
DELETE_ALL_FONTS; /* delete all downloaded fonts */
font_count = 0; /* no font numbers are assigned yet */
font_switched = TRUE;
}
/*-->devterm*/
/**********************************************************************/
/****************************** devterm *******************************/
/**********************************************************************/
void
devterm() /* terminate device */
{
DELETE_ALL_FONTS; /* delete all downloaded fonts */
RESET_PRINTER; /* printer reset */
}
#include "dvifile.h"
#include "dviinit.h"
#include "dviterm.h"
/*-->eopact*/
/**********************************************************************/
/******************************* eopact *******************************/
/**********************************************************************/
void
eopact() /* end of page action */
{
if (copies > 1)
SETCOPIES(copies);
PAGEEJECT;
}
#include "f20open.h"
#include "fatal.h"
/*-->fillrect*/
/**********************************************************************/
/****************************** fillrect ******************************/
/**********************************************************************/
void
fillrect(x,y,width,height)
COORDINATE x,y,width,height; /* lower left corner, size */
/***********************************************************************
With the page origin (0,0) at the lower-left corner, draw a filled
rectangle at (x,y).
For most TeX uses, rules are uncommon, and little optimization is
possible. However, for the LaTeX Bezier option, curves are simulated by
many small rules (typically 2 x 2) separated by positioning commands.
By remembering the size of the last rule set, we can test for the
occurrence of repeated rules of the same size, and reduce the output by
omitting the rule sizes. The last rule parameters are reset by the
begin-page action in prtpage(), so they do not carry across pages.
It is not possible to use relative, instead of absolute, moves in these
sequences, without stacking rules for the whole page, because each rule
is separated in the DVI file by push, pop, and positioning commands,
making for an uncertain correspondence between internal (xcp,ycp)
pixel page coordinates and external device coordinates.
The last string y coordinate, str_ycp, must be reset here to force any
later setstring() to reissue new absolute positioning commands.
***********************************************************************/
{
str_ycp = -1; /* invalidate string y coordinate */
if ((height != rule_height) || (width != rule_width))
{
RULE(x,y,width,height);
rule_width = width; /* save current rule parameters */
rule_height = height;
}
else /* same size rule again */
RULE2(x,y);
}
#include "findpost.h"
#include "fixpos.h"
#include "fontfile.h"
#include "fontsub.h"
#include "getbytes.h"
#include "getfntdf.h"
#include "getpgtab.h"
#include "inch.h"
#include "initglob.h"
/*-->loadbmap*/
/**********************************************************************/
/****************************** loadbmap ******************************/
/**********************************************************************/
void
loadbmap(c) /* load big character as raster bitmap */
register BYTE c;
{
register struct char_entry *tcharptr; /* temporary char_entry pointer */
void (*charyy)(); /* subterfuge to get around PCC-20 bug */
if (fontptr != pfontptr)
(void)openfont(fontptr->n);
if (fontfp == (FILE *)NULL) /* do nothing if no font file */
return;
tcharptr = &(fontptr->ch[c]);
moveto(hh - tcharptr->xoffp,YSIZE-vv+tcharptr->yoffp);
MOVETO(xcp,ycp);
BEGIN_RASTER_GRAPHICS;
/* Bug fix: PCC-20 otherwise jumps to charxx instead of *charxx */
charyy = fontptr->charxx;
(void)(*charyy)(c,outraster); /* load rasters into device */
END_RASTER_GRAPHICS;
str_ycp = -1; /* invalidate string y coordinate */
}
/*-->loadchar*/
/**********************************************************************/
/****************************** loadchar ******************************/
/**********************************************************************/
void
loadchar(c)
register BYTE c;
{
void (*charyy)(); /* subterfuge to get around PCC-20 bug */
register struct char_entry *tcharptr; /* temporary char_entry pointer */
COORDINATE nbytes; /* number of bytes per row */
float temp;
INT16 ntemp;
if ((c < FIRSTPXLCHAR) || (LASTPXLCHAR < c)) /* check character range */
return;
tcharptr = &(fontptr->ch[c]);
if (!VISIBLE(tcharptr))
return; /* do nothing for invisible fonts */
if (fontptr != pfontptr)
openfont(fontptr->n);
if (fontfp == (FILE *)NULL) /* do nothing if no font file */
return;
(void)clearerr(plotfp);
tcharptr->isloaded = TRUE;
SET_CHAR_CODE(fontptr->font_number,c);
/* Number of bytes in bitmap, possibly reduced because of LaserJet Plus */
/* limits. Large characters really need to be handled as bitmaps */
/* instead of as downloaded characters. */
nbytes = ((tcharptr->wp) + 7) >> 3; /* wp div 8 */
DOWNLOAD_CHARACTER(MIN(16,nbytes)*MIN(255,(tcharptr->hp)));
OUTC(4); /* 4 0 14 1 is fixed sequence */
OUTC(0);
OUTC(14);
OUTC(1);
OUTC(0); /* orientation = 0 ==> portrait */
OUTC(0);
/* Apologies for the temporary variables; the Sun OS 3.2 cc could
not compile the original expressions. */
ntemp = MIN(-(tcharptr->xoffp),127);
ntemp = MAX(-128,ntemp);
OUT16(ntemp);
ntemp = MIN(tcharptr->yoffp,127);
ntemp = MAX(-128,ntemp);
OUT16(ntemp);
ntemp = MIN(128,tcharptr->wp);
OUT16(ntemp);
ntemp = MIN(255,tcharptr->hp);
OUT16(ntemp);
temp = (float)(tcharptr->tfmw);
temp = 4.0*temp*conv;
ntemp = ROUND(temp);
#ifdef MT905
ntemp = ((ntemp+2)/4)*4; /* MTP fix for nearest dot */
#endif
OUT16(ntemp); /* delta x to nearest 1/4 dot */
/* Bug fix: PCC-20 otherwise jumps to charxx instead of *charxx */
charyy = fontptr->charxx;
(void)(*charyy)(c,outrow); /* output rasters */
if (DISKFULL(plotfp))
(void)fatal("loadchar(): Output error -- disk storage probably full");
}
/*-->makefont*/
/**********************************************************************/
/***************************** makefont *******************************/
/**********************************************************************/
void
makefont()
{
register UNSIGN16 the_char; /* loop index */
INT16 j; /* loop index */
/* parameters of largest cell */
COORDINATE cell_above; /* dots above baseline */
COORDINATE cell_below; /* dots below baseline */
COORDINATE char_below; /* dots below baseline */
COORDINATE cell_height; /* total height */
COORDINATE cell_left; /* dots left of character cell */
COORDINATE cell_width; /* total width */
COORDINATE cell_baseline; /* in 0..(cell_height-1) */
/*******************************************************************
Each LaserJet Plus font must be assigned a unique number in
0..32767, and only 32 fonts can be active at one time. We keep a
table of font pointers for this purpose and store the corresponding
table index in font_number. Each time a new font is encountered, we
increment font_count and store it in that font's font_number. TeX
produces a unique font number as well, and TeX82 only uses values in
the range 0..255. The DVI file supports 32-bit font numbers, so it
is better to generate our own, because someday other programs may
produce DVI files too.
If fonts are freed dynamically, the table entry must be invalidated
and a command sent to the printer to delete that font. At present,
font freeing happens only in dviterm() at the close of processing of
a single DVI file. A subsequent DVI file processed will result in
the invocation of devinit() which resets font_count to 0. It is not
possible to avoid downloading some fonts for subsequent files,
because we have no control over what order they are printed in, and
each output file must therefore be regarded as independent.
In order to deal with the limit of MAXFONTS active fonts in the
LaserJet Plus, we have two choices. The first and best and most
difficult, is to entirely discard the matching of TeX fonts with
LaserJet Plus fonts---new characters would simply be assigned to the
next available entry in the current open font. The second choice is
less satisfactory---when MAXFONTS fonts have been downloaded, we
delete the last of them and reset the isloaded flags for all the
characters in that font. The Plus also allows deletion of
individual characters, so in the first case, we could delete
little-used characters to make room in the font tables on the
device; we do not use that approach here. One problem is that font
metrics must be assigned which match the largest character in the
font, so individual character deletion and later re-use of the
vacated slot is not in general possible. A second problem is the
bizarre behavior of the LaserJet Plus in response to a font
deletion--the current page is printed, and the all text following
the delete font command will appear on the next page.
There is a silly restriction that only MAXPAGEFONTS fonts can be
used on a single page, where MAXPAGEFONTS < MAXFONTS. We handle
this by keeping a count, page_fonts, of the number of fonts in use
on the current page, and whenever a font is requested whose
font_number (0,1,...) reaches or exceeds MAXPAGEFONTS, then
setchar() and setstr() will not download the character as a font
character, but instead will set it as a raster bitmap, in the same
fashion that large characters are handled.
We handle the font deletion misfeature by just not using it. If a
document requires more than MAXFONTS fonts, then the characters for
those additional fonts are sent as raster bitmaps, instead of
downloaded characters.
These two actions completely remove any limitation on the size or
number of fonts in the LaserJet Plus.
As an optimization to reduce the effort of font table searching,
each time a font is requested which has already been loaded, we move
that font to the front of font_table[]; the least popular fonts will
therefore be the ones that get deleted. The least recently used
priority mechanism is widely used in operating systems for buffer
management; the overhead of list rearrangement here is acceptable,
because font changing occurs relatively infrequently, compared to
character setting.
<------wp------>
^ ................
| ................
| ................
| ................
| ................
| ................
| ................
hp ................
| ................
| ................
| ................
| .....o.......... <-- character reference point (xcp,ycp) at "o"
| ................ ^
| ................ |
| ................ |--- (hp - yoffp - 1)
| ................ |
v +............... v <-- (xcorner,ycorner) at "+"
<--->
|
|
xoffp (negative if (xcp,ycp) left of bitmap)
The LaserJet Plus font mechanism requires the declaration of cell
height, width, and baseline in the CreateFont command. Experiments
show that these must be large enough to incorporate the maximum
extent above and below the baseline of any characters in the font.
The metric correspondences are as follows:
----- -------------
TeX LaserJet Plus
----- -------------
xoffp -(left offset)
yoffp top offset
max(yoffp) + 1 cell baseline
max(wp) cell width
max(yoffp) +
max(hp-yoff,yoff-hp) + 1 cell height
*******************************************************************/
if (font_count >= MAXFONTS) /* then no more fonts can be downloaded */
{
/* set up so BIGCHAR() recognizes the font as not downloadable
and sends bitmaps instead */
fontptr->font_number = font_count;
for (the_char = FIRSTPXLCHAR; the_char <= LASTPXLCHAR; ++the_char)
{
fontptr->ch[the_char].isloaded = FALSE;
fontptr->ch[the_char].istoobig = TRUE;
}
return;
}
for (j = 0; j < (INT16)font_count; ++j)
if (font_table[j] == fontptr) /* then font already known */
{
for (; j > 0; --j) /* LRU: move find to front */
font_table[j] = font_table[j-1];
font_table[0] = fontptr;
return; /* nothing more to do */
}
if (j >= (INT16)font_count) /* new font */
{
fontptr->font_number = font_count;
font_table[font_count] = fontptr;
font_count++; /* count the next font */
cell_above = 0;
cell_below = 0;
cell_left = 0;
cell_width = 0;
for (the_char = FIRSTPXLCHAR; the_char <= LASTPXLCHAR; ++the_char)
{
char_below = fontptr->ch[the_char].yoffp -
fontptr->ch[the_char].hp;
char_below = ABS(char_below);
cell_height = 1 + fontptr->ch[the_char].yoffp + char_below;
fontptr->ch[the_char].isloaded = FALSE;
fontptr->ch[the_char].istoobig = (BOOLEAN)(
!IN(-128,-fontptr->ch[the_char].xoffp,127) ||
!IN(-128, fontptr->ch[the_char].yoffp,127) ||
!IN(1,fontptr->ch[the_char].wp,255) ||
!IN(1,cell_height,255));
cell_above = MAX(cell_above,fontptr->ch[the_char].yoffp);
cell_below = MAX(cell_below,char_below);
cell_left = MAX(cell_left,-fontptr->ch[the_char].xoffp);
cell_width = MAX(cell_width,fontptr->ch[the_char].wp);
}
cell_baseline = cell_above + 1;
cell_height = cell_above + cell_below + 1;
if ((cell_width > 255) || (cell_height > 255))
{ /* don't worry, big characters are handled specially later */
cell_height = MIN(cell_height,255);
cell_width = MIN(cell_width,255);
}
cell_width = MAX(1,cell_width);
SET_FONT_ID(fontptr->font_number);
CREATE_FONT;
OUTC(0);
OUTC(26); /* descriptor length */
OUTC(0); /* always 0 */
OUTC(1); /* 8-bit font (for more than 95 characters) */
OUT16(0); /* always 0 */
OUT16(cell_baseline); /* baseline position (in 0..cell_height-1)*/
OUT16(cell_width); /* cell width */
OUT16(cell_height); /* cell height */
OUTC(0); /* orientation = 0 ==> portrait */
OUTC(1); /* spacing = 1 ==> proportional */
OUT16(0); /* symbol set (arbitrary) */
OUT16(2); /* pitch (arbitrary in 2..1260) */
OUT16(cell_height); /* font height (arbitrary in 0..10922) */
OUT16(0); /* always 0 */
OUTC(0); /* always 0 */
OUTC(0); /* style = 0 ==> upright */
OUTC(0); /* stroke weight = 0 ==> normal */
OUTC(0); /* typeface = 0 (arbitrary) */
}
}
#include "movedown.h"
#include "moveover.h"
#include "moveto.h"
/*-->newfont*/
/**********************************************************************/
/****************************** newfont *******************************/
/**********************************************************************/
void
newfont()
{
/*******************************************************************
Creating fonts immediately (usually at postamble read time, before
any pages have been printed), would exhaust the LaserJet Plus font
supply in a document with more than 32 fonts, requiring many fonts
to be sent as bitmaps, instead of as downloaded characters. If only
a few pages were selected for printing, it is unlikely that more
than 32 would actually be needed. With character bitmaps, the
output file is much larger than when downloaded characters are used,
and there is a likelihood of raising printer error code 21 (page too
complex to process), at least on the LaserJet Plus, which has rather
limited internal memory (about 512Kb). The HPLJ Series II can have
up to 4.5Mb of memory, and possibly would eliminate this error
condition.
Instead, we simply mark the font as currently undefined. When it is
first referenced, setfont() will call makefont() to actually create
it in the printer.
*******************************************************************/
fontptr->font_number = UNDEFINED_FONT;
}
#include "nosignex.h"
#include "openfont.h"
#include "option.h"
/*-->outraster*/
/**********************************************************************/
/***************************** outraster ******************************/
/**********************************************************************/
void
outraster(c,yoff)/* print single raster line in bitmap graphics mode */
BYTE c; /* current character value */
UNSIGN16 yoff; /* offset from top row (0,1,...,hp-1) */
{
UNSIGN16 bytes_per_row; /* number of raster bytes to copy */
register UNSIGN16 k; /* loop index */
register UNSIGN32 *p; /* pointer into img_row[] */
struct char_entry *tcharptr;/* temporary char_entry pointer */
register BYTE the_byte; /* unpacked raster byte */
tcharptr = &(fontptr->ch[c]);/* assume check for valid c has been done */
bytes_per_row = (UNSIGN16)((tcharptr->wp) + 7) >> 3; /* wp div 8 */
p = img_row; /* we step pointer p along img_row[] */
TRANSFER_RASTER_GRAPHICS(bytes_per_row);
for (k = bytes_per_row; k > 0; ++p)
{
the_byte = (BYTE)((*p) >> 24);
OUTC(the_byte);
if ((--k) <= 0)
break;
the_byte = (BYTE)((*p) >> 16);
OUTC(the_byte);
if ((--k) <= 0)
break;
the_byte = (BYTE)((*p) >> 8);
OUTC(the_byte);
if ((--k) <= 0)
break;
the_byte = (BYTE)(*p);
OUTC(the_byte);
if ((--k) <= 0)
break;
}
}
/*-->outrow*/
/**********************************************************************/
/******************************* outrow *******************************/
/**********************************************************************/
void
outrow(c,yoff) /* copy img_row[] into rasters[] if allocated, else no-op */
BYTE c; /* current character value */
UNSIGN16 yoff; /* offset from top row (0,1,...,hp-1) */
{
UNSIGN16 bytes_per_row; /* number of raster bytes to copy */
register UNSIGN16 k; /* loop index */
register UNSIGN32 *p; /* pointer into img_row[] */
struct char_entry *tcharptr;/* temporary char_entry pointer */
register BYTE the_byte; /* unpacked raster byte */
if (yoff > 255) /* LaserJet Plus cannot handle big characters */
return;
tcharptr = &(fontptr->ch[c]);/* assume check for valid c has been done */
bytes_per_row = (UNSIGN16)((tcharptr->wp) + 7) >> 3; /* wp div 8 */
bytes_per_row = MIN(16,bytes_per_row); /* limited to 128 bits/row */
p = img_row; /* we step pointer p along img_row[] */
for (k = bytes_per_row; k > 0; ++p)
{
the_byte = (BYTE)((*p) >> 24);
OUTC(the_byte);
if ((--k) <= 0)
break;
the_byte = (BYTE)((*p) >> 16);
OUTC(the_byte);
if ((--k) <= 0)
break;
the_byte = (BYTE)((*p) >> 8);
OUTC(the_byte);
if ((--k) <= 0)
break;
the_byte = (BYTE)(*p);
OUTC(the_byte);
if ((--k) <= 0)
break;
}
}
#include "prtpage.h"
#include "readfont.h"
#include "readgf.h"
#include "readpk.h"
#include "readpost.h"
#include "readpxl.h"
#include "reldfont.h"
#include "rulepxl.h"
/*-->setchar*/
/**********************************************************************/
/****************************** setchar *******************************/
/**********************************************************************/
void
setchar(c, update_h)
register BYTE c;
register BOOLEAN update_h;
{
register struct char_entry *tcharptr; /* temporary char_entry pointer */
/* BIGCHAR() and ONPAGE() are used here and in setstr(). BIGCHAR()
is TRUE if the character is too big to send as a downloaded font, or
if more than MAXPAGEFONTS have been used on this page, and the
required font's font_number is not in the page_members set, or if
more the font is not in the first MAXFONTS fonts. */
#define BIGCHAR(t) (((t->wp > (COORDINATE)size_limit) ||\
(t->hp > (COORDINATE)size_limit)) || (fontptr->font_number >= MAXFONTS) ||\
(t->istoobig) ||\
((page_fonts > MAXPAGEFONTS) && !IS_MEMBER(page_members,fontptr->font_number)))
#define ONPAGE(t) (((hh - t->xoffp + t->pxlw) <= XSIZE) \
&& (hh >= 0)\
&& (vv <= YSIZE)\
&& (vv >= 0))
if (DBGOPT(DBG_SET_TEXT))
{
(void)fprintf(stderr,"setchar('");
if (isprint(c))
(void)putc(c,stderr);
else
(void)fprintf(stderr,"\\%03o",(int)c);
(void)fprintf(stderr,"'<%d>) (hh,vv) = (%ld,%ld) font name <%s>",
(int)c, (long)hh, (long)vv, fontptr->n);
NEWLINE(stderr);
}
tcharptr = &(fontptr->ch[c]);
moveto(hh,(COORDINATE)(YSIZE-vv));
if (ONPAGE(tcharptr))
{ /* character fits entirely on page */
if (font_switched)
{
(void)setfont();
font_switched = FALSE;
}
if (VISIBLE(tcharptr))
{
if (BIGCHAR(tcharptr))
loadbmap(c);
else
{
if (!tcharptr->isloaded)
loadchar(c);
if (ycp != str_ycp)
{
MOVETO(xcp,ycp);
str_ycp = ycp;
}
else
MOVEX(xcp);
OUT_CHAR(c);
}
}
}
else if (DBGOPT(DBG_OFF_PAGE) && !quiet)
{ /* character is off page -- discard it */
(void)fprintf(stderr,
"setchar(): Char %c [10#%3d 8#%03o 16#%02x] off page.",
isprint(c) ? c : '?',c,c,c);
NEWLINE(stderr);
}
if (update_h)
{
h += (INT32)tcharptr->tfmw;
hh += (COORDINATE)tcharptr->pxlw;
hh = fixpos(hh-lmargin,h,conv) + lmargin;
}
}
#include "setfntnm.h"
/*-->setfont*/
/**********************************************************************/
/****************************** setfont *******************************/
/**********************************************************************/
void
setfont()
{
if (fontptr->font_number == UNDEFINED_FONT)
(void)makefont();
if (!IS_MEMBER(page_members,fontptr->font_number))
page_fonts++;
if (page_fonts <= MAXPAGEFONTS)
ADD_MEMBER(page_members,fontptr->font_number);
if (!((fontptr->font_number >= MAXFONTS) || ((page_fonts > MAXPAGEFONTS) &&
!IS_MEMBER(page_members,fontptr->font_number))))
{
if (fontptr->font_number)
OUTF("\033\050%dX",fontptr->font_number);
else
OUTS("\033\050X");
}
}
#include "setrule.h"
/*-->setstr*/
/**********************************************************************/
/******************************* setstr *******************************/
/**********************************************************************/
void
setstr(c)
register BYTE c;
{
register struct char_entry *tcharptr; /* temporary char_entry pointer */
register BOOLEAN inside;
INT32 h0,v0; /* (h,v) at entry */
COORDINATE hh0,vv0; /* (hh,vv) at entry */
COORDINATE hh_last; /* hh before call to fixpos() */
register UNSIGN16 k; /* loop index */
UNSIGN16 nstr; /* number of characters in str[] */
BYTE str[MAXSTR+1]; /* string accumulator */
BOOLEAN truncated; /* off-page string truncation flag */
/*******************************************************************
Set a sequence of characters in SETC_000 .. SETC_127 with a minimal
number of LaserJet Plus commands. These sequences tend to occur in
long clumps in a DVI file, and setting them together whenever
possible substantially decreases the overhead and the size of the
output file. A sequence can be set as a single string if
* TeX and LaserJet Plus coordinates of each character agree (may
not be true, since device coordinates are in multiples of 1/4
pixel; violation of this requirement can be detected if
fixpos() changes hh, or if ycp != str_ycp), AND
* each character is in the same font (this will always be true
in a sequence from a DVI file), AND
* each character fits within the page boundaries, AND
* each character definition is already loaded, AND
* each character is from a visible font, AND
* each character bitmap extent is smaller than the size_limit
(which is used to enable discarding large characters after
each use in order to conserve virtual memory storage on the
output device).
Whenever any of these conditions does not hold, any string already
output is terminated, and a new one begun.
Two output optimizations are implemented here. First, up to MAXSTR
(in practice more than enough) characters are collected in str[],
and any that require downloading are handled. Then the entire
string is set at once, subject to the above limitations. Second, by
recording the vertical page coordinate, ycp, in the global variable
str_ycp (reset in prtpage() at begin-page processing), it is
possible to avoid outputting y coordinates unnecessarily, since a
single line of text will generally result in many calls to this
function.
*******************************************************************/
#define BEGINSTRING {inside = TRUE;\
if (ycp != str_ycp)\
{\
MOVETO(xcp,ycp);\
str_ycp = ycp;\
}\
else\
MOVEX(xcp);}
#define ENDSTRING {inside = FALSE;}
#define OFF_PAGE (-1) /* off-page coordinate value */
if (font_switched) /* output new font selection */
{
(void)setfont();
font_switched = FALSE;
}
inside = FALSE;
truncated = FALSE;
hh0 = hh;
vv0 = vv;
h0 = h;
v0 = v;
nstr = 0;
while ((SETC_000 <= c) && (c <= SETC_127) && (nstr < MAXSTR))
{ /* loop over character sequence */
tcharptr = &(fontptr->ch[c]);
moveto(hh,YSIZE-vv);
if (ONPAGE(tcharptr) && VISIBLE(tcharptr))
{ /* character fits entirely on page and is visible */
if ((!tcharptr->isloaded) && (!BIGCHAR(tcharptr)))
loadchar(c);
}
/* update horizontal positions in TFM and pixel units */
h += (INT32)tcharptr->tfmw;
hh += (COORDINATE)tcharptr->pxlw;
str[nstr++] = c; /* save string character */
c = (BYTE)nosignex(dvifp,(BYTE)1);
}
/* put back character which terminated the loop */
(void)UNGETC((int)(c),dvifp);
hh = hh0; /* restore coordinates at entry */
vv = vv0;
h = h0;
v = v0;
if (DBGOPT(DBG_SET_TEXT))
{
(void)fprintf(stderr,"setstr(\"");
for (k = 0; k < nstr; ++k)
{
c = str[k];
if (isprint(c))
(void)putc(c,stderr);
else
(void)fprintf(stderr,"\\%03o",(int)c);
}
(void)fprintf(stderr,"\") (hh,vv) = (%ld,%ld) font name <%s>",
(long)hh, (long)vv, fontptr->n);
NEWLINE(stderr);
}
for (k = 0; k < nstr; ++k)
{ /* loop over character sequence */
c = str[k];
tcharptr = &(fontptr->ch[c]);
moveto(hh,YSIZE-vv);
if (ONPAGE(tcharptr) && VISIBLE(tcharptr))
{ /* character fits entirely on page and is visible */
/* We must check first for a big character. The character */
/* may have been downloaded on a previous page, but if it is */
/* not in the page_members font set, we must send it again */
/* as a bitmap instead. */
if (BIGCHAR(tcharptr))
{
if (inside)
ENDSTRING; /* finish any open string */
loadbmap(c);
}
else /* character already downloaded */
{
if (!inside)
BEGINSTRING;
OUT_CHAR(c);
}
}
else /* character does not fit on page -- output */
{ /* current string and discard the character */
truncated = TRUE;
if (inside)
ENDSTRING;
}
/* update horizontal positions in TFM and pixel units */
h += (INT32)tcharptr->tfmw;
hh += (COORDINATE)tcharptr->pxlw;
hh_last = hh;
hh = fixpos(hh-lmargin,h,conv) + lmargin;
if (DBGOPT(DBG_POS_CHAR))
{
(void)fprintf(stderr,
"[%03o] xcp = %d\tycp = %d\thh = %d\thh_last = %d\n",
c,xcp,ycp,hh,hh_last);
}
/* If fixpos() changed position, we need new string next time */
/* around. Actually, since the LaserJet Plus stores character */
/* widths in units of 1/4 dot, we could compute coordinates at */
/* four times the precision, but for now, we start a new string */
/* each time we have one or more dots of error. */
if ((hh != hh_last) && inside)
ENDSTRING;
}
if (truncated && DBGOPT(DBG_OFF_PAGE) && !quiet)
{
(void)fprintf(stderr,"setstr(): Text [");
for (k = 0; k < nstr; ++k)
(void)fprintf(stderr,isprint(str[k]) ? "%c" : "\\%03o",str[k]);
(void)fprintf(stderr,"] truncated at page boundaries.");
NEWLINE(stderr);
}
if (inside) /* finish last string */
ENDSTRING;
}
#include "signex.h"
#include "skgfspec.h"
#include "skipfont.h"
#include "skpkspec.h"
#include "special.h"
#include "strchr.h"
#include "strcm2.h"
#include "strid2.h"
#include "strrchr.h"
#include "tctos.h"
#include "usage.h"
#include "warning.h"